import random
import re
from datetime import datetime

from flask import abort, jsonify
from flask import current_app
from flask import make_response
from flask import request
from  flask  import  session
from info import constants, redis_store, db
from info.libs.yuntongxun.sms import CCP
from info.models import User
from info.utils.response_code import RET
from  . import  passport_blu
from  info.utils.captcha.captcha import captcha


@passport_blu.route('/logout')
def logout():
    """
    退出登陆

    :return:
    """

    # pop是移除session 中的数据
    # pop 会有一个返回值,如果要移除的key不存在, 就返回None
    session.pop('user_id', None)
    session.pop('mobile', None)
    session.pop('nich_name', None)

    # is_admin的值如果不清除，先登陆管理员,会保存到session 再登陆普通用户，又能访问管理员页面
    session.pop('is_admin', None)


    return  jsonify(errno = RET.OK,errmsg = "退出成功")





@passport_blu.route('/login',methods=["POST"])
def login():
    """
    登陆代码

    1.获取参数
    2.校验参数
    3.校验密码是否正确
    4.保存用户的登陆状态
    5.响应
    :return:
    """

    #  1.获取参数
    params_dict = request.json

    mobile = params_dict.get('mobile')
    passport = params_dict.get('passport')

    # 2.校验参数
    if not all([mobile,passport]):
        return  jsonify(errno=RET.PARAMERR,errmsg= "参数错误")

    #  校验手机号是否正确
    if not re.match('1[35678]\\d{9}',mobile):
        return  jsonify(errno=RET.PARAMERR,errmsg = "手机号格式错误")

    # 3.校验密码是否正确
    # 先查询当前是否有指定手机号的用户
    try:
        user = User.query.filter(User.mobile ==mobile).first()

    except Exception as e:
        current_app.logger.error(e)
        return  jsonify(errno = RET.DBERR,errmsg = "数据查询错误")
    # 判断用户是否存在
    if not user:
        return  jsonify(errno = RET.NODATA,errmsg = "用户不存在")

    # 校验登陆的密码核当前用户的密码是否一致
    if not user.check_passowrd(passport):
        return  jsonify(errno = RET.PWDERR,errmsg = "用户名或密码错误")


    # 4.保存用户的登陆状态
    session["user_id"] = user.id
    session["mobile"] = user.mobile
    session["nick_name"] = user.nick_name

    # 设置当前用户最后一次的登陆时间
    user.last_login = datetime.now()

    # try:
    #     db.session.commit()
    # except Exception as e :
    #     db.session.rollback()
    #     current_app.logger.error(e)


    #  5.响应
    return  jsonify(errno = RET.OK,errmsg = "登陆成功")





@passport_blu.route('/register',methods = ["POST"])
def register():
    """
    注册的逻辑
    1.获取参数
    2.校验参数
    3..取到服务器保存的真实的短信验证码内容
    4.校验用户输入的短信验证码内容和很真实验证码内容是否一致
    5.如果一致,初始化 User 模型,并且赋值属性
    6.将user 模型添加到数据库
    7. 返回响应
    :return:
    """
    #1.获取参数
    param_dict = request.json
    mobile = param_dict.get("mobile")
    smscode = param_dict.get("smscode")
    password = param_dict.get("password")

    #2.校验参数
    if not all([mobile,smscode,password]):
        return  jsonify(errno =RET.PARAMERR, errmsg = "参数")

    # 校验手机号是否正确
    if not re.match('1[35678]\\d{9}',mobile):
        return  jsonify(errno = RET.PARAMERR,errmsg = "手机号格式不正确")

    #3.取到服务器保存的真实的短信验证码内容
    try:
        real_sms_code = redis_store.get("SMS_" + mobile)
    except Exception as e :
        current_app.logger.error(e)
        return  jsonify(errno = RET.DBERR,errmsg = "数据查询失败")

    if  not  real_sms_code:
        return  jsonify(errno = RET.NODATA, errmsg = "验证码已过期")
    # 4.校验用户输入的短信验证码内容和很真实验证码内容是否一致
    if real_sms_code  != smscode:
        return  jsonify(errno = RET.DATAERR,errmsg = "验证码输入错误")

     #5.如果一致,初始化 User 模型,并且赋值属性
    user = User()
    user.mobile = mobile
    # 暂时没有昵称,使用手机号代替
    user.nick_name = mobile
    # 记录用户最后一次登陆时间
    user.last_login = datetime.now()

    # 对密码做处理

    user.password = password

     # 6.将user 模型添加到数据库

    try:
        db.session.add(user)
        db.session.commit()
    except Exception  as  e:
        current_app.logger.error(e)
        db.session.rollback()
        return  jsonify(errno=RET.DBERR,errmsg = "数据保存失败")


    # 往 session 中 保存数据表示当前已经登陆
    session["user_id"] = user.id
    session["mobile"] = user.mobile
    session["nick_name"] = user.nick_name



    #7. 返回响应
    return  jsonify(errno = RET.OK,errmsg = "注册成功")








@passport_blu.route('/sms_code',methods = ["POST",'GET'])
def send_sms_code():
    """
    发送短信的逻辑
    1.获取参数: 手机号,图片验证码内容,图片验证码的编号(随机值)
    2.校验参数(参数是否符合规则,判断是否有值)
    3.先从redis 中 取出真实的验证码内容
    4.与用户的验证码内容进行对比,如果对比不一致,那么返回验证码输入错误
    5.如果一致,生成验证码的内容(随机数据)
    6.发送短信验证码
    7.告发发送结果
    :return:
    """

    # 1.获取参数
    params_dict = request.json
    mobile = params_dict.get("mobile")
    image_code = params_dict.get("image_code")
    image_code_id = params_dict.get("image_code_id")

    # 2.校验参数
    if not all([mobile,image_code,image_code_id]):
        return  jsonify(errno = RET.PARAMERR,errmsg ='参数有误')

    if  not re.match('1[35678]\\d{9}',mobile):
        print(1111)
        return  jsonify(errno = RET.PARAMERR,errmsg = "手机号格式不正确")

    # 3. 先从redis  中 取出真实的验证码内容

    try:
        real_image_code = redis_store.get("ImageCodeId_" + image_code_id)

    except Exception as e:
        current_app.logger.error(e)
        return  jsonify(errno = RET.DBERR, errmsg ="数据查询失败")

    if  not real_image_code:
        return  jsonify(errno = RET.NODATA, errmsq = "图片验证码已过期")


    # 4.与用户的验证码内容进行对比, 如果不一致,那么返回验证码输入错误
    if real_image_code.upper() != image_code.upper():

        return  jsonify(errno = RET.DATAERR,errmsg = '验证码输入错误')


    # 5.如果一直,生成短信验证码的内容(随机数据)
    # 随机数字,保证数字长度为6位,不够在前面补上0
    sms_code_str = "%06d" % random.randint(0,999999)
    current_app.logger.debug("短信验证码内容是: %s" % sms_code_str)
    print("短信验证码是", sms_code_str)
    # 6.发送短信验证码
    result = CCP().send_template_sms(mobile,[sms_code_str,constants.SMS_CODE_REDIS_EXPIRES / 60 ],"1")
    if result != 0:
        # 代表发送不成功
         return  jsonify(errno = RET.THIRDERR,errmsg = "发送短信失败")

    # 保存验证码内容到redis
    try:
        redis_store.set("SMS_" + mobile, sms_code_str,constants.SMS_CODE_REDIS_EXPIRES)
    except  Exception  as e :
        current_app.logger.error(e)
        return  jsonify(errno = RET.DBERR,errmsg = "数据保存失败")

    # 7. 告知发送结果
    return  jsonify(errno = RET.OK, errmsg ="发送成功" )





@passport_blu.route('/image_code')
def get_image_code():
    """
    生成图片验证码并返回
    1.取到参数
    2.判断参数是否有值
    3.生成图片验证码
    4.保存图片验证文字内容到redis
    5.返回验证码图片
    :return:
    """

    # 1.取到参数
    # args: 取到url  中 ? 后面的参数
    image_code_id=request.args.get("imageCodeId",None)

    # 2.判断参数是否有值
    if not image_code_id:
        return  abort(403)

    # 3.生成图片验证码
    name,text,image = captcha.generate_captcha()


    # 4.保存图片验证码文字内容到redis
    try:
        redis_store.set("ImageCodeId_" + image_code_id,text,constants.IMAGE_CODE_REDIS_EXPIRES)

    except Exception  as e:
        current_app.logger.error(e)
        abort(500)

    # 5.返回验证码图片
    response = make_response(image)
    print("图片验证码是", text)
    # 设置数据的类型  以便浏览器更加智能 识别其是什么类型
    response.headers["Content-Type"] = "image/jpg"
    return  response